/*
 * FindBugs - Find bugs in Java programs
 * Copyright (C) 2004-2006 University of Maryland
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

package edu.umd.cs.findbugs.detect;

import java.util.HashSet;
import java.util.Set;

import org.apache.bcel.classfile.Code;
import org.apache.bcel.classfile.JavaClass;

import edu.umd.cs.findbugs.BugReporter;
import edu.umd.cs.findbugs.NonReportingDetector;
import edu.umd.cs.findbugs.OpcodeStack;
import edu.umd.cs.findbugs.ProgramPoint;
import edu.umd.cs.findbugs.ba.AnalysisContext;
import edu.umd.cs.findbugs.ba.FieldSummary;
import edu.umd.cs.findbugs.ba.Hierarchy2;
import edu.umd.cs.findbugs.ba.XClass;
import edu.umd.cs.findbugs.ba.XField;
import edu.umd.cs.findbugs.ba.XMethod;
import edu.umd.cs.findbugs.bcel.OpcodeStackDetector;
import edu.umd.cs.findbugs.visitclass.PreorderVisitor;

public class FieldItemSummary extends OpcodeStackDetector implements NonReportingDetector {
	
	FieldSummary fieldSummary = new FieldSummary();
	public FieldItemSummary(BugReporter bugReporter) {
		AnalysisContext context = AnalysisContext.currentAnalysisContext();
		context.setFieldSummary(fieldSummary);
	}

	Set<XField> touched = new HashSet<XField>();

	
	@Override
	public boolean shouldVisit(JavaClass obj) {
		return !getXClass().hasStubs();
	}
	boolean sawInitializeSuper;
	@Override
	public void sawOpcode(int seen) {
		if (getMethodName().equals("<init>") && seen == INVOKEVIRTUAL) {
			XMethod m = getXMethodOperand();
			if (m != null && !m.isPrivate() && !m.isFinal()) {
			int args = PreorderVisitor.getNumberArguments(m.getSignature());
			OpcodeStack.Item item = stack.getStackItem(args);
			if (item.getRegisterNumber() == 0) {
				try {
                	Set<XMethod> targets = Hierarchy2.resolveVirtualMethodCallTargets(m, false, false);
	                for(XMethod called : targets) {
	                	if (!called.isAbstract() && !called.equals(m))
	                		fieldSummary.setCalledFromSuperConstructor(new ProgramPoint(this), called);
	                }
                } catch (ClassNotFoundException e) {
	               AnalysisContext.reportMissingClass(e);
                }
				
			}
			
			}
		
		}
		
		if (seen == INVOKESPECIAL && getMethodName().equals("<init>") && getNameConstantOperand().equals("<init>")) {
			
			String classOperand = getClassConstantOperand();
			OpcodeStack.Item invokedOn = stack.getItemMethodInvokedOn(this);
			if (invokedOn.getRegisterNumber() == 0 && !classOperand.equals(getClassName())) {
				sawInitializeSuper = true;
				XMethod invoked = getXMethodOperand();
				if (invoked != null)
					fieldSummary.sawSuperCall(getXMethod(), invoked);
			}

		}

		if (seen == PUTFIELD || seen == PUTSTATIC) {
			XField fieldOperand = getXFieldOperand();
			if (fieldOperand == null) return;
			touched.add(fieldOperand);
			if (!fieldOperand.getClassDescriptor().getClassName().equals(getClassName()))
				fieldSummary.addWrittenOutsideOfConstructor(fieldOperand);
			else if (seen == PUTFIELD) {
				OpcodeStack.Item addr = stack.getStackItem(1); {
				if (addr.getRegisterNumber() != 0 || !getMethodName().equals("<init>"))
					fieldSummary.addWrittenOutsideOfConstructor(fieldOperand);
				}
			} else if (seen == PUTSTATIC && !getMethodName().equals("<clinit>"))
				fieldSummary.addWrittenOutsideOfConstructor(fieldOperand);
			OpcodeStack.Item top = stack.getStackItem(0);
			fieldSummary.mergeSummary(fieldOperand, top);
		}

	}
		
		 
		@Override
        public void visit(Code obj) {
			sawInitializeSuper = false;
			super.visit(obj);
			fieldSummary.setFieldsWritten(getXMethod(), touched);
			if (getMethodName().equals("<init>") && sawInitializeSuper) {
				XClass thisClass = getXClass();
				for(XField f : thisClass.getXFields()) 
					if (!touched.contains(f)) {
					OpcodeStack.Item item;
					char firstChar = f.getSignature().charAt(0);
					if (firstChar == 'L' || firstChar == '[')
						item = OpcodeStack.Item.nullItem(f.getSignature());
					else if (firstChar == 'I')
						item = new OpcodeStack.Item("I", (Integer) 0);
					else if (firstChar == 'J')
						item = new OpcodeStack.Item("J", (Long) 0L);
					else
						item = new OpcodeStack.Item(f.getSignature());
					fieldSummary.mergeSummary(f, item);
					}
			}
			touched.clear();
		}
		
		
		@Override
        public void report() {
			fieldSummary.setComplete(true);
		}

}
